home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
CD ROM Paradise Collection 4
/
CD ROM Paradise Collection 4 1995 Nov.iso
/
graphics
/
povray2.zip
/
SOURCE
/
TRIANGLE.C
< prev
next >
Wrap
C/C++ Source or Header
|
1993-10-25
|
20KB
|
695 lines
/****************************************************************************
* triangle.c
*
* This module implements primitives for triangles and smooth triangles.
*
* from Persistence of Vision Raytracer
* Copyright 1993 Persistence of Vision Team
*---------------------------------------------------------------------------
* NOTICE: This source code file is provided so that users may experiment
* with enhancements to POV-Ray and to port the software to platforms other
* than those supported by the POV-Ray Team. There are strict rules under
* which you are permitted to use this file. The rules are in the file
* named POVLEGAL.DOC which should be distributed with this file. If
* POVLEGAL.DOC is not available or for more info please contact the POV-Ray
* Team Coordinator by leaving a message in CompuServe's Graphics Developer's
* Forum. The latest version of POV-Ray may be found there as well.
*
* This program is based on the popular DKB raytracer version 2.12.
* DKBTrace was originally written by David K. Buck.
* DKBTrace Ver 2.0-2.12 were written by David K. Buck & Aaron A. Collins.
*
*****************************************************************************/
#include "frame.h"
#include "vector.h"
#include "povproto.h"
METHODS Triangle_Methods =
{
All_Triangle_Intersections,
Inside_Triangle, Triangle_Normal,
Copy_Triangle,
Translate_Triangle, Rotate_Triangle,
Scale_Triangle, Transform_Triangle, Invert_Triangle, Destroy_Triangle
};
METHODS Smooth_Triangle_Methods =
{
All_Triangle_Intersections,
Inside_Triangle, Smooth_Triangle_Normal,
Copy_Smooth_Triangle,
Translate_Smooth_Triangle, Rotate_Smooth_Triangle,
Scale_Smooth_Triangle, Transform_Smooth_Triangle,
Invert_Smooth_Triangle, Destroy_Triangle
};
extern RAY *CM_Ray;
extern long Ray_Triangle_Tests, Ray_Triangle_Tests_Succeeded;
#define max3(x,y,z) ((x>y)?((x>z)?1:3):((y>z)?2:3))
#define MAX3(x,y,z) (((x)>(y))?(((x)>(z))?(x):(z)):(((y)>(z))?(y):(z)))
#define MIN3(x,y,z) (((x)<(y))?(((x)<(z))?(x):(z)):(((y)<(z))?(y):(z)))
void Find_Triangle_Dominant_Axis(Triangle)
TRIANGLE *Triangle;
{
DBL x, y, z;
x = fabs(Triangle->Normal_Vector.x);
y = fabs (Triangle->Normal_Vector.y);
z = fabs (Triangle->Normal_Vector.z);
switch (max3(x, y, z))
{
case 1:
Triangle->Dominant_Axis = X_AXIS;
break;
case 2:
Triangle->Dominant_Axis = Y_AXIS;
break;
case 3:
Triangle->Dominant_Axis = Z_AXIS;
break;
}
}
void Compute_Smooth_Triangle (Triangle)
SMOOTH_TRIANGLE *Triangle;
{
VECTOR P3MinusP2, VTemp1, VTemp2;
DBL x, y, z, uDenominator, Proj;
VSub (P3MinusP2, Triangle->P3, Triangle->P2);
x = fabs (P3MinusP2.x);
y = fabs (P3MinusP2.y);
z = fabs (P3MinusP2.z);
switch (max3 (x, y, z))
{
case 1:
Triangle->vAxis = X_AXIS;
Triangle->BaseDelta = P3MinusP2.x;
break;
case 2:
Triangle->vAxis = Y_AXIS;
Triangle->BaseDelta = P3MinusP2.y;
break;
case 3:
Triangle->vAxis = Z_AXIS;
Triangle->BaseDelta = P3MinusP2.z;
break;
}
VSub (VTemp1, Triangle->P2, Triangle->P3);
VNormalize (VTemp1, VTemp1);
VSub (VTemp2, Triangle->P1, Triangle->P3);
VDot (Proj, VTemp2, VTemp1);
VScaleEq (VTemp1, Proj);
VSub (Triangle->Perp, VTemp1, VTemp2);
VNormalize (Triangle->Perp, Triangle->Perp);
VDot (uDenominator, VTemp2, Triangle->Perp);
VInverseScaleEq (Triangle->Perp, -uDenominator);
}
int Compute_Triangle (Triangle,Smooth)
TRIANGLE *Triangle;
int Smooth;
{
VECTOR V1, V2, Temp;
DBL Length, T1, T2, T3;
VSub (V1, Triangle->P1, Triangle->P2);
VSub (V2, Triangle->P3, Triangle->P2);
VCross (Triangle->Normal_Vector, V1, V2);
VLength (Length, Triangle->Normal_Vector);
/* Set up a flag so we can ignore degenerate triangles */
if (Length < 1.0e-9)
{
Triangle->Degenerate_Flag = TRUE;
return (0);
}
/* Normalize the normal vector. */
VInverseScaleEq (Triangle->Normal_Vector, Length);
VDot (Triangle->Distance, Triangle->Normal_Vector, Triangle->P1);
Triangle->Distance *= -1.0;
Find_Triangle_Dominant_Axis(Triangle);
switch (Triangle->Dominant_Axis)
{
case X_AXIS:
if ((Triangle->P2.y - Triangle->P3.y)*(Triangle->P2.z - Triangle->P1.z) <
(Triangle->P2.z - Triangle->P3.z)*(Triangle->P2.y - Triangle->P1.y))
{
Temp = Triangle->P2;
Triangle->P2 = Triangle->P1;
Triangle->P1 = Temp;
if (Smooth)
{
Temp = ((SMOOTH_TRIANGLE *) Triangle)->N2;
((SMOOTH_TRIANGLE *) Triangle)->N2 = ((SMOOTH_TRIANGLE *) Triangle)->N1;
((SMOOTH_TRIANGLE *) Triangle)->N1 = Temp;
}
}
break;
case Y_AXIS:
if ((Triangle->P2.x - Triangle->P3.x)*(Triangle->P2.z - Triangle->P1.z) <
(Triangle->P2.z - Triangle->P3.z)*(Triangle->P2.x - Triangle->P1.x))
{
Temp = Triangle->P2;
Triangle->P2 = Triangle->P1;
Triangle->P1 = Temp;
if (Smooth)
{
Temp = ((SMOOTH_TRIANGLE *) Triangle)->N2;
((SMOOTH_TRIANGLE *) Triangle)->N2 = ((SMOOTH_TRIANGLE *) Triangle)->N1;
((SMOOTH_TRIANGLE *) Triangle)->N1 = Temp;
}
}
break;
case Z_AXIS:
if ((Triangle->P2.x - Triangle->P3.x)*(Triangle->P2.y - Triangle->P1.y) <
(Triangle->P2.y - Triangle->P3.y)*(Triangle->P2.x - Triangle->P1.x))
{
Temp = Triangle->P2;
Triangle->P2 = Triangle->P1;
Triangle->P1 = Temp;
if (Smooth)
{
Temp = ((SMOOTH_TRIANGLE *) Triangle)->N2;
((SMOOTH_TRIANGLE *) Triangle)->N2 = ((SMOOTH_TRIANGLE *) Triangle)->N1;
((SMOOTH_TRIANGLE *) Triangle)->N1 = Temp;
}
}
break;
}
if (Smooth)
Compute_Smooth_Triangle((SMOOTH_TRIANGLE *) Triangle);
/* Build the bounding information from the vertices */
/* Use temps so macro not too big. */
T1 = MIN3(Triangle->P1.x, Triangle->P2.x, Triangle->P3.x);
T2 = MIN3(Triangle->P1.y, Triangle->P2.y, Triangle->P3.y);
T3 = MIN3(Triangle->P1.z, Triangle->P2.z, Triangle->P3.z);
Make_Vector(&Triangle->Bounds.Lower_Left,T1,T2,T3);
T1 = MAX3(Triangle->P1.x, Triangle->P2.x, Triangle->P3.x);
T2 = MAX3(Triangle->P1.y, Triangle->P2.y, Triangle->P3.y);
T3 = MAX3(Triangle->P1.z, Triangle->P2.z, Triangle->P3.z);
Make_Vector(&Triangle->Bounds.Lengths,T1,T2,T3);
VSub(Triangle->Bounds.Lengths, Triangle->Bounds.Lengths,
Triangle->Bounds.Lower_Left);
Triangle->Bounds.Lengths.x += EPSILON;
Triangle->Bounds.Lengths.y += EPSILON;
Triangle->Bounds.Lengths.z += EPSILON;
return (1);
}
int All_Triangle_Intersections (Object, Ray, Depth_Stack)
OBJECT *Object;
RAY *Ray;
ISTACK *Depth_Stack;
{
DBL Depth;
VECTOR IPoint;
if (Intersect_Triangle (Ray, (TRIANGLE *)Object, &Depth))
{
VScale (IPoint, Ray -> Direction, Depth);
VAddEq (IPoint, Ray -> Initial);
if (Point_In_Clip(&IPoint,Object->Clip))
{
push_entry(Depth,IPoint,Object,Depth_Stack);
return (TRUE);
}
}
return (FALSE);
}
int Intersect_Triangle (Ray, Triangle, Depth)
RAY *Ray;
TRIANGLE *Triangle;
DBL *Depth;
{
DBL NormalDotOrigin, NormalDotDirection;
DBL s, t;
Ray_Triangle_Tests++;
if(Triangle->Degenerate_Flag)
return(FALSE);
if (Ray == CM_Ray)
{
if (!Triangle->CMCached)
{
VDot (Triangle->CMNormDotOrigin, Triangle->Normal_Vector, Ray->Initial);
Triangle->CMNormDotOrigin += Triangle->Distance;
Triangle->CMNormDotOrigin *= -1.0;
Triangle->CMCached = TRUE;
}
VDot (NormalDotDirection, Triangle->Normal_Vector, Ray->Direction);
if ((NormalDotDirection < Small_Tolerance) &&
(NormalDotDirection > -Small_Tolerance))
return (FALSE);
*Depth = Triangle->CMNormDotOrigin / NormalDotDirection;
}
else
{
VDot (NormalDotOrigin, Triangle->Normal_Vector, Ray->Initial);
NormalDotOrigin += Triangle->Distance;
NormalDotOrigin *= -1.0;
VDot (NormalDotDirection, Triangle->Normal_Vector, Ray->Direction);
if ((NormalDotDirection < Small_Tolerance) &&
(NormalDotDirection > -Small_Tolerance))
return (FALSE);
*Depth = NormalDotOrigin / NormalDotDirection;
}
if ((*Depth < Small_Tolerance) || (*Depth > Max_Distance))
return (FALSE);
switch (Triangle->Dominant_Axis)
{
case X_AXIS:
s = Ray->Initial.y + *Depth * Ray->Direction.y;
t = Ray->Initial.z + *Depth * Ray->Direction.z;
if ((Triangle->P2.y - s)*(Triangle->P2.z - Triangle->P1.z) <
(Triangle->P2.z - t)*(Triangle->P2.y - Triangle->P1.y))
return (FALSE);
if ((Triangle->P3.y - s)*(Triangle->P3.z - Triangle->P2.z) <
(Triangle->P3.z - t)*(Triangle->P3.y - Triangle->P2.y))
return (FALSE);
if ((Triangle->P1.y - s)*(Triangle->P1.z - Triangle->P3.z) <
(Triangle->P1.z - t)*(Triangle->P1.y - Triangle->P3.y))
return (FALSE);
Ray_Triangle_Tests_Succeeded++;
return (TRUE);
case Y_AXIS:
s = Ray->Initial.x + *Depth * Ray->Direction.x;
t = Ray->Initial.z + *Depth * Ray->Direction.z;
if ((Triangle->P2.x - s)*(Triangle->P2.z - Triangle->P1.z) <
(Triangle->P2.z - t)*(Triangle->P2.x - Triangle->P1.x))
return (FALSE);
if ((Triangle->P3.x - s)*(Triangle->P3.z - Triangle->P2.z) <
(Triangle->P3.z - t)*(Triangle->P3.x - Triangle->P2.x))
return (FALSE);
if ((Triangle->P1.x - s)*(Triangle->P1.z - Triangle->P3.z) <
(Triangle->P1.z - t)*(Triangle->P1.x - Triangle->P3.x))
return (FALSE);
Ray_Triangle_Tests_Succeeded++;
return (TRUE);
case Z_AXIS:
s = Ray->Initial.x + *Depth * Ray->Direction.x;
t = Ray->Initial.y + *Depth * Ray->Direction.y;
if ((Triangle->P2.x - s)*(Triangle->P2.y - Triangle->P1.y) <
(Triangle->P2.y - t)*(Triangle->P2.x - Triangle->P1.x))
return (FALSE);
if ((Triangle->P3.x - s)*(Triangle->P3.y - Triangle->P2.y) <
(Triangle->P3.y - t)*(Triangle->P3.x - Triangle->P2.x))
return (FALSE);
if ((Triangle->P1.x - s)*(Triangle->P1.y - Triangle->P3.y) <
(Triangle->P1.y - t)*(Triangle->P1.x - Triangle->P3.x))
return (FALSE);
Ray_Triangle_Tests_Succeeded++;
return (TRUE);
}
return (FALSE);
}
int Inside_Triangle (IPoint, Object)
VECTOR *IPoint;
OBJECT *Object;
{
return (FALSE);
}
void Triangle_Normal (Result, Object, IPoint)
OBJECT *Object;
VECTOR *Result, *IPoint;
{
*Result = ((TRIANGLE *)Object)->Normal_Vector;
}
void *Copy_Triangle (Object)
OBJECT *Object;
{
TRIANGLE *New;
New = Create_Triangle ();
*New = * ((TRIANGLE *)Object);
return (New);
}
void Translate_Triangle (Object, Vector)
OBJECT *Object;
VECTOR *Vector;
{
TRIANGLE *Triangle = (TRIANGLE *) Object;
VECTOR Translation;
if(Triangle->Degenerate_Flag) return;
VEvaluate (Translation, Triangle->Normal_Vector, *Vector);
Triangle->Distance -= Translation.x + Translation.y + Translation.z;
VAddEq (Triangle->P1, *Vector)
VAddEq (Triangle->P2, *Vector)
VAddEq (Triangle->P3, *Vector)
/* Recalculate the bounds */
VAddEq(Object->Bounds.Lower_Left, *Vector);
}
void Rotate_Triangle (Object, Vector)
OBJECT *Object;
VECTOR *Vector;
{
TRANSFORM Trans;
if(((TRIANGLE *)Object)->Degenerate_Flag) return;
Compute_Rotation_Transform (&Trans, Vector);
Transform_Triangle (Object, &Trans);
}
void Scale_Triangle (Object, Vector)
OBJECT *Object;
VECTOR *Vector;
{
TRIANGLE *Triangle = (TRIANGLE *) Object;
DBL Length,T1,T2,T3;
if(Triangle->Degenerate_Flag) return;
Triangle->Normal_Vector.x = Triangle->Normal_Vector.x / Vector->x;
Triangle->Normal_Vector.y = Triangle->Normal_Vector.y / Vector->y;
Triangle->Normal_Vector.z = Triangle->Normal_Vector.z / Vector->z;
VLength(Length, Triangle->Normal_Vector);
VInverseScaleEq (Triangle->Normal_Vector, Length);
Triangle->Distance /= Length;
VEvaluateEq (Triangle->P1, *Vector);
VEvaluateEq (Triangle->P2, *Vector);
VEvaluateEq (Triangle->P3, *Vector);
/* Recompute the bounds */
/* Use temps so macro not too big. */
T1 = MIN3(Triangle->P1.x, Triangle->P2.x, Triangle->P3.x);
T2 = MIN3(Triangle->P1.y, Triangle->P2.y, Triangle->P3.y);
T3 = MIN3(Triangle->P1.z, Triangle->P2.z, Triangle->P3.z);
Make_Vector(&Triangle->Bounds.Lower_Left,T1,T2,T3);
T1 = MAX3(Triangle->P1.x, Triangle->P2.x, Triangle->P3.x);
T2 = MAX3(Triangle->P1.y, Triangle->P2.y, Triangle->P3.y);
T3 = MAX3(Triangle->P1.z, Triangle->P2.z, Triangle->P3.z);
Make_Vector(&Triangle->Bounds.Lengths,T1,T2,T3);
VSub(Triangle->Bounds.Lengths, Triangle->Bounds.Lengths,
Triangle->Bounds.Lower_Left);
Triangle->Bounds.Lengths.x += EPSILON;
Triangle->Bounds.Lengths.y += EPSILON;
Triangle->Bounds.Lengths.z += EPSILON;
}
void Transform_Triangle (Object, Trans)
OBJECT *Object;
TRANSFORM *Trans;
{
TRIANGLE *Triangle = (TRIANGLE *) Object;
if(Triangle->Degenerate_Flag) return;
MTransPoint (&Triangle->Normal_Vector,
&Triangle->Normal_Vector, Trans);
MTransPoint (&Triangle->P1, &Triangle->P1, Trans);
MTransPoint (&Triangle->P2, &Triangle->P2, Trans);
MTransPoint (&Triangle->P3, &Triangle->P3, Trans);
Compute_Triangle (Triangle,FALSE);
}
TRIANGLE *Create_Triangle()
{
TRIANGLE *New;
if ((New = (TRIANGLE *) malloc (sizeof (TRIANGLE))) == NULL)
MAError ("triangle");
INIT_OBJECT_FIELDS(New,TRIANGLE_OBJECT,&Triangle_Methods)
Make_Vector (&(New -> Normal_Vector), 0.0, 1.0, 0.0);
New -> Distance = 0.0;
New -> CMNormDotOrigin = 0.0;
New -> CMCached = FALSE;
Make_Vector (&(New -> P1), 0.0, 0.0, 0.0);
Make_Vector (&(New -> P2), 1.0, 0.0, 0.0);
Make_Vector (&(New -> P3), 0.0, 1.0, 0.0);
New -> Degenerate_Flag = FALSE;
/* NOTE: Dominant_Axis is computed when Parse_Triangle calls
Compute_Triangle. vAxis is used only for smooth triangles */
return (New);
}
void Invert_Triangle (Object)
OBJECT *Object;
{
return;
}
/* Calculate the Phong-interpolated vector within the triangle
at the given intersection point. The math for this is a bit
bizarre:
- P1
| /|\ \
| / |Perp\
| / V \ \
| / | \ \
u | /____|_____PI___\
| / | \ \
- P2-----|--------|----P3
Pbase PIntersect
|-------------------|
v
Triangle->Perp is a unit vector from P1 to Pbase. We calculate
u = (PI - P1) DOT Perp / ((P3 - P1) DOT Perp).
We then calculate where the line from P1 to PI intersects the line P2 to P3:
PIntersect = (PI - P1)/u.
We really only need one coordinate of PIntersect. We then calculate v as:
v = PIntersect.x / (P3.x - P2.x)
or v = PIntersect.y / (P3.y - P2.y)
or v = PIntersect.z / (P3.z - P2.z)
depending on which calculation will give us the best answers.
Once we have u and v, we can perform the normal interpolation as:
NTemp1 = N1 + u(N2 - N1);
NTemp2 = N1 + u(N3 - N1);
Result = normalize (NTemp1 + v(NTemp2 - NTemp1))
As always, any values which are constant for the triangle are cached
in the triangle.
*/
void Smooth_Triangle_Normal (Result, Object, IPoint)
OBJECT *Object;
VECTOR *Result, *IPoint;
{
SMOOTH_TRIANGLE *Triangle = (SMOOTH_TRIANGLE *) Object;
VECTOR PIMinusP1, NTemp1, NTemp2;
DBL u = 0.0, v = 0.0;
VSub (PIMinusP1, *IPoint, Triangle->P1);
VDot (u, PIMinusP1, Triangle->Perp);
if (u < 1.0e-9)
{
*Result = Triangle->N1;
return;
}
/* BaseDelta contains P3.x-P2.x, P3.y-P2.y, or P3.z-P2.z depending on the
value of vAxis. */
switch (Triangle->vAxis)
{
case X_AXIS:
v = (PIMinusP1.x/u + Triangle->P1.x - Triangle->P2.x) / Triangle->BaseDelta;
break;
case Y_AXIS:
v = (PIMinusP1.y/u + Triangle->P1.y - Triangle->P2.y) / Triangle->BaseDelta;
break;
case Z_AXIS:
v = (PIMinusP1.z/u + Triangle->P1.z - Triangle->P2.z)/ Triangle->BaseDelta;
break;
}
VSub (NTemp1, Triangle->N2, Triangle->N1);
VScaleEq (NTemp1, u);
VAddEq (NTemp1, Triangle->N1);
VSub (NTemp2, Triangle->N3, Triangle->N1);
VScaleEq (NTemp2, u);
VAddEq (NTemp2, Triangle->N1);
VSub (*Result, NTemp2, NTemp1);
VScaleEq (*Result, v);
VAddEq (*Result, NTemp1);
VNormalize (*Result, *Result);
}
void *Copy_Smooth_Triangle (Object)
OBJECT *Object;
{
SMOOTH_TRIANGLE *New;
New = Create_Smooth_Triangle ();
*New = * ((SMOOTH_TRIANGLE *)Object);
return (New);
}
void Translate_Smooth_Triangle (Object, Vector)
OBJECT *Object;
VECTOR *Vector;
{
SMOOTH_TRIANGLE *Triangle = (SMOOTH_TRIANGLE *) Object;
VECTOR Translation;
if(Triangle->Degenerate_Flag) return;
VEvaluate (Translation, Triangle->Normal_Vector, *Vector);
Triangle->Distance -= Translation.x + Translation.y + Translation.z;
VAddEq (Triangle->P1, *Vector)
VAddEq (Triangle->P2, *Vector)
VAddEq (Triangle->P3, *Vector)
Compute_Triangle ((TRIANGLE *) Triangle,TRUE);
}
void Rotate_Smooth_Triangle (Object, Vector)
OBJECT *Object;
VECTOR *Vector;
{
TRANSFORM Trans;
if(((SMOOTH_TRIANGLE *)Object)->Degenerate_Flag) return;
Compute_Rotation_Transform (&Trans, Vector);
Transform_Smooth_Triangle (Object, &Trans);
}
void Scale_Smooth_Triangle (Object, Vector)
OBJECT *Object;
VECTOR *Vector;
{
SMOOTH_TRIANGLE *Triangle = (SMOOTH_TRIANGLE *) Object;
DBL Length;
if(Triangle->Degenerate_Flag) return;
Triangle->Normal_Vector.x = Triangle->Normal_Vector.x / Vector->x;
Triangle->Normal_Vector.y = Triangle->Normal_Vector.y / Vector->y;
Triangle->Normal_Vector.z = Triangle->Normal_Vector.z / Vector->z;
VLength(Length, Triangle->Normal_Vector);
VScaleEq (Triangle->Normal_Vector, 1.0 / Length);
Triangle->Distance /= Length;
VEvaluateEq (Triangle->P1, *Vector);
VEvaluateEq (Triangle->P2, *Vector);
VEvaluateEq (Triangle->P3, *Vector);
Compute_Triangle ((TRIANGLE *) Triangle,TRUE);
}
void Transform_Smooth_Triangle (Object, Trans)
OBJECT *Object;
TRANSFORM *Trans;
{
SMOOTH_TRIANGLE *Triangle = (SMOOTH_TRIANGLE *) Object;
if(Triangle->Degenerate_Flag) return;
MTransPoint (&Triangle->Normal_Vector,
&Triangle->Normal_Vector, Trans);
MTransPoint (&Triangle->P1, &Triangle->P1, Trans);
MTransPoint (&Triangle->P2, &Triangle->P2, Trans);
MTransPoint (&Triangle->P3, &Triangle->P3, Trans);
MTransPoint (&Triangle->N1, &Triangle->N1, Trans);
MTransPoint (&Triangle->N2, &Triangle->N2, Trans);
MTransPoint (&Triangle->N3, &Triangle->N3, Trans);
Compute_Triangle ((TRIANGLE *) Triangle,TRUE);
}
void Invert_Smooth_Triangle (Object)
OBJECT *Object;
{
return;
}
SMOOTH_TRIANGLE *Create_Smooth_Triangle()
{
SMOOTH_TRIANGLE *New;
if ((New = (SMOOTH_TRIANGLE *) malloc (sizeof (SMOOTH_TRIANGLE))) == NULL)
MAError ("smooth triangle");
INIT_OBJECT_FIELDS(New,SMOOTH_TRIANGLE_OBJECT,&Smooth_Triangle_Methods)
Make_Vector (&(New->Normal_Vector), 0.0, 1.0, 0.0);
New->Distance = 0.0;
New -> CMNormDotOrigin = 0.0;
New -> CMCached = FALSE;
Make_Vector (&(New -> P1), 0.0, 0.0, 0.0);
Make_Vector (&(New -> P2), 1.0, 0.0, 0.0);
Make_Vector (&(New -> P3), 0.0, 1.0, 0.0);
Make_Vector (&(New -> N1), 0.0, 1.0, 0.0);
Make_Vector (&(New -> N2), 0.0, 1.0, 0.0);
Make_Vector (&(New -> N3), 0.0, 1.0, 0.0);
New -> BaseDelta = 0.0;
New -> Degenerate_Flag = FALSE;
/* NOTE: Dominant_Axis and vAxis are computed when Parse_Triangle calls
Compute_Triangle. */
return (New);
}
void Destroy_Triangle (Object)
OBJECT *Object;
{
free (Object);
}